home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / BROWSER.XPI / bin / components / nsHelperAppDlg.js < prev    next >
Encoding:
Text File  |  2006-09-10  |  33.8 KB  |  873 lines

  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
  2. /* ***** BEGIN LICENSE BLOCK *****
  3.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  4.  *
  5.  * The contents of this file are subject to the Mozilla Public License Version
  6.  * 1.1 (the "License"); you may not use this file except in compliance with
  7.  * the License. You may obtain a copy of the License at
  8.  * http://www.mozilla.org/MPL/
  9.  *
  10.  * Software distributed under the License is distributed on an "AS IS" basis,
  11.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  12.  * for the specific language governing rights and limitations under the
  13.  * License.
  14.  *
  15.  * The Original Code is the Mozilla browser.
  16.  *
  17.  * The Initial Developer of the Original Code is
  18.  * Netscape Communications Corporation.
  19.  * Portions created by the Initial Developer are Copyright (C) 2001
  20.  * the Initial Developer. All Rights Reserved.
  21.  *
  22.  * Contributor(s):
  23.  *   Bill Law    <law@netscape.com>
  24.  *   Scott MacGregor <mscott@netscape.com>
  25.  *
  26.  * Alternatively, the contents of this file may be used under the terms of
  27.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  28.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  29.  * in which case the provisions of the GPL or the LGPL are applicable instead
  30.  * of those above. If you wish to allow use of your version of this file only
  31.  * under the terms of either the GPL or the LGPL, and not to allow others to
  32.  * use your version of this file under the terms of the MPL, indicate your
  33.  * decision by deleting the provisions above and replace them with the notice
  34.  * and other provisions required by the GPL or the LGPL. If you do not delete
  35.  * the provisions above, a recipient may use your version of this file under
  36.  * the terms of any one of the MPL, the GPL or the LGPL.
  37.  *
  38.  * ***** END LICENSE BLOCK ***** */
  39.  
  40. /* This file implements the nsIHelperAppLauncherDialog interface.
  41.  *
  42.  * The implementation consists of a JavaScript "class" named nsHelperAppDialog,
  43.  * comprised of:
  44.  *   - a JS constructor function
  45.  *   - a prototype providing all the interface methods and implementation stuff
  46.  *
  47.  * In addition, this file implements an nsIModule object that registers the
  48.  * nsHelperAppDialog component.
  49.  */
  50.  
  51. const nsIHelperAppLauncherDialog = Components.interfaces.nsIHelperAppLauncherDialog;
  52. const REASON_CANTHANDLE = nsIHelperAppLauncherDialog.REASON_CANTHANDLE;
  53. const REASON_SERVERREQUEST = nsIHelperAppLauncherDialog.REASON_SERVERREQUEST;
  54. const REASON_TYPESNIFFED = nsIHelperAppLauncherDialog.REASON_TYPESNIFFED;
  55.  
  56.  
  57. /* ctor
  58.  */
  59. function nsHelperAppDialog() {
  60.     // Initialize data properties.
  61.     this.mLauncher = null;
  62.     this.mContext  = null;
  63.     this.mSourcePath = null;
  64.     this.chosenApp = null;
  65.     this.givenDefaultApp = false;
  66.     this.strings   = new Array;
  67.     this.elements  = new Array;
  68.     this.updateSelf = true;
  69.     this.mTitle    = "";
  70.     this.mIsMac    = false;
  71. }
  72.  
  73. nsHelperAppDialog.prototype = {
  74.     // Turn this on to get debugging messages.
  75.     debug: false,
  76.  
  77.     nsIMIMEInfo  : Components.interfaces.nsIMIMEInfo,
  78.  
  79.     // Dump text (if debug is on).
  80.     dump: function( text ) {
  81.         if ( this.debug ) {
  82.             dump( text );
  83.         }
  84.     },
  85.  
  86.     // This "class" supports nsIHelperAppLauncherDialog, and nsISupports.
  87.     QueryInterface: function (iid) {
  88.         if (iid.equals(Components.interfaces.nsIHelperAppLauncherDialog) ||
  89.             iid.equals(Components.interfaces.nsISupports))
  90.             return this;
  91.  
  92.         Components.returnCode = Components.results.NS_ERROR_NO_INTERFACE;
  93.         return null;
  94.     },
  95.  
  96.     // ---------- nsIHelperAppLauncherDialog methods ----------
  97.  
  98.     // show: Open XUL dialog using window watcher.  Since the dialog is not
  99.     //       modal, it needs to be a top level window and the way to open
  100.     //       one of those is via that route).
  101.     show: function(aLauncher, aContext, aReason)  {
  102.          this.mLauncher = aLauncher;
  103.          this.mContext  = aContext;
  104.          this.mReason   = aReason;
  105.          // Display the dialog using the Window Watcher interface.
  106.          var ww = Components.classes["@mozilla.org/embedcomp/window-watcher;1"]
  107.                     .getService( Components.interfaces.nsIWindowWatcher );
  108.          this.mDialog = ww.openWindow( null, // no parent
  109.                                        "chrome://global/content/nsHelperAppDlg.xul",
  110.                                        null,
  111.                                        "chrome,titlebar,dialog=yes",
  112.                                        null );
  113.          // Hook this object to the dialog.
  114.          this.mDialog.dialog = this;
  115.          // Watch for error notifications.
  116.          this.mIsMac = (this.mDialog.navigator.platform.indexOf( "Mac" ) != -1);
  117.          this.progressListener.helperAppDlg = this;
  118.          this.mLauncher.setWebProgressListener( this.progressListener );
  119.     },
  120.  
  121.     // promptForSaveToFile:  Display file picker dialog and return selected file.
  122.     promptForSaveToFile: function(aLauncher, aContext, aDefaultFile, aSuggestedFileExtension) {
  123.         var result = "";
  124.  
  125.         const prefSvcContractID = "@mozilla.org/preferences-service;1";
  126.         const prefSvcIID = Components.interfaces.nsIPrefService;
  127.         var branch = Components.classes[prefSvcContractID].getService(prefSvcIID)
  128.                                                           .getBranch("browser.download.");
  129.         var dir = null;
  130.  
  131.         const nsILocalFile = Components.interfaces.nsILocalFile;
  132.         const kDownloadDirPref = "dir";
  133.  
  134.         // Try and pull in download directory pref
  135.         try {
  136.             dir = branch.getComplexValue(kDownloadDirPref, nsILocalFile);
  137.         } catch (e) { }
  138.  
  139.         var bundle = Components.classes["@mozilla.org/intl/stringbundle;1"]
  140.                                .getService(Components.interfaces.nsIStringBundleService)
  141.                                .createBundle("chrome://global/locale/nsHelperAppDlg.properties");
  142.  
  143.         var autoDownload = branch.getBoolPref("autoDownload");
  144.         // If the autoDownload pref is set then just download to default download directory
  145.         if (autoDownload && dir && dir.exists()) {
  146.             if (aDefaultFile == "")
  147.                 aDefaultFile = bundle.GetStringFromName("noDefaultFile") + (aSuggestedFileExtension || "");
  148.             dir.append(aDefaultFile);
  149.             return uniqueFile(dir);
  150.         }
  151.  
  152.         // Use file picker to show dialog.
  153.         var nsIFilePicker = Components.interfaces.nsIFilePicker;
  154.         var picker = Components.classes[ "@mozilla.org/filepicker;1" ]
  155.                        .createInstance( nsIFilePicker );
  156.  
  157.         var windowTitle = bundle.GetStringFromName( "saveDialogTitle" );
  158.  
  159.         var parent = aContext
  160.                         .QueryInterface( Components.interfaces.nsIInterfaceRequestor )
  161.                         .getInterface( Components.interfaces.nsIDOMWindowInternal );
  162.         picker.init( parent, windowTitle, nsIFilePicker.modeSave );
  163.         picker.defaultString = aDefaultFile;
  164.         if (aSuggestedFileExtension) {
  165.             // aSuggestedFileExtension includes the period, so strip it
  166.             picker.defaultExtension = aSuggestedFileExtension.substring(1);
  167.         } else {
  168.             try {
  169.                 picker.defaultExtension = this.mLauncher.MIMEInfo.primaryExtension;
  170.             } catch (ex) {
  171.             }
  172.         }
  173.  
  174.         var wildCardExtension = "*";
  175.         if ( aSuggestedFileExtension ) {
  176.             wildCardExtension += aSuggestedFileExtension;
  177.             picker.appendFilter( wildCardExtension, wildCardExtension );
  178.         }
  179.  
  180.         picker.appendFilters( nsIFilePicker.filterAll );
  181.  
  182.         try {
  183.             if (dir.exists())
  184.                 picker.displayDirectory = dir;
  185.         } catch (e) { }
  186.  
  187.         if (picker.show() == nsIFilePicker.returnCancel || !picker.file) {
  188.             // Null result means user cancelled.
  189.             return null;
  190.         }
  191.  
  192.         // If not using specified location save the user's choice of directory
  193.         if (branch.getBoolPref("lastLocation") || autoDownload) {
  194.             var directory = picker.file.parent.QueryInterface(nsILocalFile);
  195.             branch.setComplexValue(kDownloadDirPref, nsILocalFile, directory);
  196.         }
  197.  
  198.         return picker.file;
  199.     },
  200.  
  201.     // ---------- implementation methods ----------
  202.  
  203.     // Web progress listener so we can detect errors while mLauncher is
  204.     // streaming the data to a temporary file.
  205.     progressListener: {
  206.         // Implementation properties.
  207.         helperAppDlg: null,
  208.  
  209.         // nsIWebProgressListener methods.
  210.         // Look for error notifications and display alert to user.
  211.         onStatusChange: function( aWebProgress, aRequest, aStatus, aMessage ) {
  212.             if ( aStatus != Components.results.NS_OK ) {
  213.                 // Get prompt service.
  214.                 var prompter = Components.classes[ "@mozilla.org/embedcomp/prompt-service;1" ]
  215.                                    .getService( Components.interfaces.nsIPromptService );
  216.                 // Display error alert (using text supplied by back-end).
  217.                 prompter.alert( this.dialog, this.helperAppDlg.mTitle, aMessage );
  218.  
  219.                 // Close the dialog.
  220.                 this.helperAppDlg.onCancel();
  221.                 if ( this.helperAppDlg.mDialog ) {
  222.                     this.helperAppDlg.mDialog.close();
  223.                 }
  224.             }
  225.         },
  226.  
  227.         // Ignore onProgressChange, onStateChange, onLocationChange, and onSecurityChange notifications.
  228.         onProgressChange: function( aWebProgress,
  229.                                     aRequest,
  230.                                     aCurSelfProgress,
  231.                                     aMaxSelfProgress,
  232.                                     aCurTotalProgress,
  233.                                     aMaxTotalProgress ) {
  234.         },
  235.  
  236.         onProgressChange64: function( aWebProgress,
  237.                                       aRequest,
  238.                                       aCurSelfProgress,
  239.                                       aMaxSelfProgress,
  240.                                       aCurTotalProgress,
  241.                                       aMaxTotalProgress ) {
  242.         },
  243.  
  244.         onStateChange: function( aWebProgress, aRequest, aStateFlags, aStatus ) {
  245.         },
  246.  
  247.         onLocationChange: function( aWebProgress, aRequest, aLocation ) {
  248.         },
  249.  
  250.         onSecurityChange: function( aWebProgress, aRequest, state ) {
  251.         }
  252.     },
  253.  
  254.     // initDialog:  Fill various dialog fields with initial content.
  255.     initDialog : function() {
  256.          // Put product brand short name in prompt.
  257.          var prompt = this.dialogElement( "prompt" );
  258.          var modified = this.replaceInsert( prompt.firstChild.nodeValue, 1, this.getString( "brandShortName" ) );
  259.          prompt.firstChild.nodeValue = modified;
  260.  
  261.          // Put file name in window title.
  262.          var suggestedFileName = this.mLauncher.suggestedFileName;
  263.  
  264.          // Some URIs do not implement nsIURL, so we can't just QI.
  265.          var url = this.mLauncher.source.clone();
  266.          try {
  267.            url.userPass = "";
  268.          } catch (ex) {}
  269.          var fname = "";
  270.          this.mSourcePath = url.prePath;
  271.          try {
  272.              url = url.QueryInterface( Components.interfaces.nsIURL );
  273.              // A url, use file name from it.
  274.              fname = url.fileName;
  275.              this.mSourcePath += url.directory;
  276.          } catch (ex) {
  277.              // A generic uri, use path.
  278.              fname = url.path;
  279.              this.mSourcePath += url.path;
  280.          }
  281.  
  282.          if (suggestedFileName)
  283.            fname = suggestedFileName;
  284.  
  285.          this.mTitle = this.replaceInsert( this.mDialog.document.title, 1, fname);
  286.          this.mDialog.document.title = this.mTitle;
  287.  
  288.          // Put content type, filename and location into intro.
  289.          this.initIntro(url, fname);
  290.  
  291.          var iconString = "moz-icon://" + fname + "?size=32&contentType=" + this.mLauncher.MIMEInfo.MIMEType;
  292.  
  293.          this.dialogElement("contentTypeImage").setAttribute("src", iconString);
  294.  
  295.          this.initAppAndSaveToDiskValues();
  296.  
  297.          // Initialize "always ask me" box. This should always be disabled
  298.          // and set to true for the ambiguous type application/octet-stream.
  299.          // Same if this dialog was forced due to a server request or type
  300.          // sniffing
  301.          var alwaysHandleCheckbox = this.dialogElement( "alwaysHandle" );
  302.          if (this.mReason != REASON_CANTHANDLE ||
  303.              this.mLauncher.MIMEInfo.MIMEType == "application/octet-stream"){
  304.             alwaysHandleCheckbox.checked = false;
  305.             alwaysHandleCheckbox.disabled = true;
  306.          }
  307.          else {
  308.             alwaysHandleCheckbox.checked = !this.mLauncher.MIMEInfo.alwaysAskBeforeHandling;
  309.          }
  310.  
  311.          // Position it.
  312.          if ( this.mDialog.opener ) {
  313.              this.mDialog.moveToAlertPosition();
  314.          } else {
  315.              this.mDialog.sizeToContent();
  316.              this.mDialog.centerWindowOnScreen();
  317.          }
  318.  
  319.          // Set initial focus
  320.          this.dialogElement( "mode" ).focus();
  321.  
  322.          this.mDialog.document.documentElement.getButton("accept").disabled = true;
  323.          const nsITimer = Components.interfaces.nsITimer;
  324.          this._timer = Components.classes["@mozilla.org/timer;1"]
  325.                                  .createInstance(nsITimer);
  326.          this._timer.initWithCallback(this, 250, nsITimer.TYPE_ONE_SHOT);
  327.     },
  328.  
  329.     // initIntro:
  330.     initIntro: function(url, filename) {
  331.         var intro = this.dialogElement( "intro" );
  332.         var desc = this.mLauncher.MIMEInfo.description;
  333.  
  334.         var text;
  335.         if ( this.mReason == REASON_CANTHANDLE )
  336.           text = "intro.";
  337.         else if (this.mReason == REASON_SERVERREQUEST )
  338.           text = "intro.attachment.";
  339.         else if (this.mReason == REASON_TYPESNIFFED )
  340.           text = "intro.sniffed.";
  341.  
  342.         var modified;
  343.         if (desc)
  344.           modified = this.replaceInsert( this.getString( text + "label" ), 1, desc );
  345.         else
  346.           modified = this.getString( text + "noDesc.label" );
  347.  
  348.         modified = this.replaceInsert( modified, 2, this.mLauncher.MIMEInfo.MIMEType );
  349.         modified = this.replaceInsert( modified, 3, filename);
  350.         modified = this.replaceInsert( modified, 4, this.getString( "brandShortName" ));
  351.  
  352.         // if mSourcePath is a local file, then let's use the pretty path name instead of an ugly
  353.         // url...
  354.         var pathString = url.prePath;
  355.         try
  356.         {
  357.           var fileURL = url.QueryInterface(Components.interfaces.nsIFileURL);
  358.           if (fileURL)
  359.           {
  360.             var fileObject = fileURL.file;
  361.             if (fileObject)
  362.             {
  363.               var parentObject = fileObject.parent;
  364.               if (parentObject)
  365.               {
  366.                 pathString = parentObject.path;
  367.               }
  368.             }
  369.           }
  370.         } catch(ex) {}
  371.  
  372.  
  373.         intro.firstChild.nodeValue = "";
  374.         intro.firstChild.nodeValue = modified;
  375.  
  376.         // Set the location text, which is separate from the intro text so it can be cropped
  377.         var location = this.dialogElement( "location" );
  378.         location.value = pathString;
  379.         location.setAttribute( "tooltiptext", this.mSourcePath );
  380.     },
  381.  
  382.     _timer: null,
  383.     notify: function (aTimer) {
  384.         try { // The user may have already canceled the dialog.
  385.           if (!this._blurred)
  386.             this.mDialog.document.documentElement.getButton('accept').disabled = false;
  387.         } catch (ex) {}
  388.         this._timer = null;
  389.     },
  390.  
  391.     _blurred: false,
  392.     onBlur: function(aEvent) {
  393.         if (aEvent.target != this.mDialog.document)
  394.           return;
  395.  
  396.         this._blurred = true;
  397.         this.mDialog.document.documentElement.getButton("accept").disabled = true;
  398.     },
  399.  
  400.     onFocus: function(aEvent) {
  401.         if (aEvent.target != this.mDialog.document)
  402.           return;
  403.  
  404.         this._blurred = false;
  405.         if (!this._timer) {
  406.           // Don't enable the button if the initial timer is running
  407.           var script = "document.documentElement.getButton('accept').disabled = false";
  408.           this.mDialog.setTimeout(script, 250);
  409.         }
  410.     },
  411.  
  412.     // Returns true iff opening the default application makes sense.
  413.     openWithDefaultOK: function() {
  414.         var result;
  415.  
  416.         // The checking is different on Windows...
  417.         if ( this.mDialog.navigator.platform.indexOf( "Win" ) != -1 ) {
  418.             // Windows presents some special cases.
  419.             // We need to prevent use of "system default" when the file is
  420.             // executable (so the user doesn't launch nasty programs downloaded
  421.             // from the web), and, enable use of "system default" if it isn't
  422.             // executable (because we will prompt the user for the default app
  423.             // in that case).
  424.  
  425.             // Need to get temporary file and check for executable-ness.
  426.             var tmpFile = this.mLauncher.targetFile;
  427.  
  428.             //  Default is Ok if the file isn't executable (and vice-versa).
  429.             result = !tmpFile.isExecutable();
  430.         } else {
  431.             // On other platforms, default is Ok if there is a default app.
  432.             // Note that nsIMIMEInfo providers need to ensure that this holds true
  433.             // on each platform.
  434.             result = this.mLauncher.MIMEInfo.hasDefaultHandler;
  435.         }
  436.         return result;
  437.     },
  438.  
  439.     // Set "default" application description field.
  440.     initDefaultApp: function() {
  441.         // Use description, if we can get one.
  442.         var desc = this.mLauncher.MIMEInfo.defaultDescription;
  443.         if ( desc ) {
  444.             this.dialogElement( "useSystemDefault" ).label = this.replaceInsert( this.getString( "defaultApp" ), 1, desc );
  445.         }
  446.     },
  447.  
  448.     // getPath:
  449.     getPath: function(file) {
  450.         if (this.mIsMac) {
  451.             return file.leafName || file.path;
  452.         }
  453.  
  454.         return file.path;
  455.     },
  456.  
  457.     // initAppAndSaveToDiskValues:
  458.     initAppAndSaveToDiskValues: function() {
  459.         // Fill in helper app info, if there is any.
  460.         this.chosenApp = this.mLauncher.MIMEInfo.preferredApplicationHandler;
  461.         // Initialize "default application" field.
  462.         this.initDefaultApp();
  463.  
  464.         // Fill application name textbox.
  465.         if (this.chosenApp && this.chosenApp.path) {
  466.             this.dialogElement( "appPath" ).value = this.getPath(this.chosenApp);
  467.         }
  468.  
  469.         var useDefault = this.dialogElement( "useSystemDefault" );;
  470.         if (this.mLauncher.MIMEInfo.preferredAction == this.nsIMIMEInfo.useSystemDefault &&
  471.             this.mReason != REASON_SERVERREQUEST) {
  472.             // Open (using system default).
  473.             useDefault.radioGroup.selectedItem = useDefault;
  474.         } else if (this.mLauncher.MIMEInfo.preferredAction == this.nsIMIMEInfo.useHelperApp &&
  475.                    this.mReason != REASON_SERVERREQUEST) {
  476.             // Open with given helper app.
  477.             var openUsing = this.dialogElement( "openUsing" );
  478.             openUsing.radioGroup.selectedItem = openUsing;
  479.         } else {
  480.             // Save to disk.
  481.             var saveToDisk = this.dialogElement( "saveToDisk" );
  482.             saveToDisk.radioGroup.selectedItem = saveToDisk;
  483.         }
  484.         // If we don't have a "default app" then disable that choice.
  485.         if ( !this.openWithDefaultOK() ) {
  486.             // Disable that choice.
  487.             useDefault.hidden = true;
  488.             // If that's the default, then switch to "save to disk."
  489.             if ( useDefault.selected ) {
  490.                 useDefault.radioGroup.selectedItem = this.dialogElement( "saveToDisk" );
  491.             }
  492.         }
  493.  
  494.         // Enable/Disable choose application button and textfield
  495.         this.toggleChoice();
  496.     },
  497.  
  498.     // Enable pick app button if the user chooses that option.
  499.     toggleChoice : function () {
  500.         // See what option is selected.
  501.         var openUsing = this.dialogElement( "openUsing" ).selected;
  502.         this.dialogElement( "chooseApp" ).disabled = !openUsing;
  503.         // appPath is always disabled on Mac
  504.         this.dialogElement( "appPath" ).disabled = !openUsing || this.mIsMac;
  505.         this.updateOKButton();
  506.     },
  507.  
  508.     // Returns the user-selected application
  509.     helperAppChoice: function() {
  510.         var result = this.chosenApp;
  511.         if (!this.mIsMac) {
  512.             var typed  = this.dialogElement( "appPath" ).value;
  513.             // First, see if one was chosen via the Choose... button.
  514.             if ( result ) {
  515.                 // Verify that the user didn't type in something different later.
  516.                 if ( typed != result.path ) {
  517.                     // Use what was typed in.
  518.                     try {
  519.                         result.QueryInterface( Components.interfaces.nsILocalFile ).initWithPath( typed );
  520.                     } catch( e ) {
  521.                         // Invalid path was typed.
  522.                         result = null;
  523.                     }
  524.                 }
  525.             } else {
  526.                 // The user didn't use the Choose... button, try using what they typed in.
  527.                 result = Components.classes[ "@mozilla.org/file/local;1" ]
  528.                     .createInstance( Components.interfaces.nsILocalFile );
  529.                 try {
  530.                     result.initWithPath( typed );
  531.                 } catch( e ) {
  532.                     result = null;
  533.                 }
  534.             }
  535.             // Remember what was chosen.
  536.             this.chosenApp = result;
  537.         }
  538.         return result;
  539.     },
  540.  
  541.     updateOKButton: function() {
  542.         var ok = false;
  543.         if ( this.dialogElement( "saveToDisk" ).selected )
  544.         {
  545.             // This is always OK.
  546.             ok = true;
  547.         }
  548.         else if ( this.dialogElement( "useSystemDefault" ).selected )
  549.         {
  550.             // No app need be specified in this case.
  551.             ok = true;
  552.         }
  553.         else
  554.         {
  555.             // only enable the OK button if we have a default app to use or if
  556.             // the user chose an app....
  557.             ok = this.chosenApp || /\S/.test( this.dialogElement( "appPath" ).value );
  558.         }
  559.  
  560.         // Enable Ok button if ok to press.
  561.         this.mDialog.document.documentElement.getButton( "accept" ).disabled = !ok;
  562.     },
  563.  
  564.     // Returns true iff the user-specified helper app has been modified.
  565.     appChanged: function() {
  566.         return this.helperAppChoice() != this.mLauncher.MIMEInfo.preferredApplicationHandler;
  567.     },
  568.  
  569.     updateMIMEInfo: function() {
  570.         var needUpdate = false;
  571.         // If current selection differs from what's in the mime info object,
  572.         // then we need to update.
  573.         // However, we don't want to change the action all nsIMIMEInfo objects to
  574.         // saveToDisk if mReason is REASON_SERVERREQUEST.
  575.         if ( this.dialogElement( "saveToDisk" ).selected &&
  576.              this.mReason != REASON_SERVERREQUEST ) {
  577.             needUpdate = this.mLauncher.MIMEInfo.preferredAction != this.nsIMIMEInfo.saveToDisk;
  578.             if ( needUpdate )
  579.                 this.mLauncher.MIMEInfo.preferredAction = this.nsIMIMEInfo.saveToDisk;
  580.         } else if ( this.dialogElement( "useSystemDefault" ).selected ) {
  581.             needUpdate = this.mLauncher.MIMEInfo.preferredAction != this.nsIMIMEInfo.useSystemDefault;
  582.             if ( needUpdate )
  583.                 this.mLauncher.MIMEInfo.preferredAction = this.nsIMIMEInfo.useSystemDefault;
  584.         } else if ( this.dialogElement( "openUsing" ).selected ) {
  585.             // For "open with", we need to check both preferred action and whether the user chose
  586.             // a new app.
  587.             needUpdate = this.mLauncher.MIMEInfo.preferredAction != this.nsIMIMEInfo.useHelperApp || this.appChanged();
  588.             if ( needUpdate ) {
  589.                 this.mLauncher.MIMEInfo.preferredAction = this.nsIMIMEInfo.useHelperApp;
  590.                 // App may have changed - Update application and description
  591.                 var app = this.helperAppChoice();
  592.                 this.mLauncher.MIMEInfo.preferredApplicationHandler = app;
  593.                 this.mLauncher.MIMEInfo.applicationDescription = "";
  594.             }
  595.         }
  596.         // Only care about the state of "always ask" if this dialog wasn't forced
  597.         if ( this.mReason == REASON_CANTHANDLE )
  598.         {
  599.           // We will also need to update if the "always ask" flag has changed.
  600.           needUpdate = needUpdate || this.mLauncher.MIMEInfo.alwaysAskBeforeHandling == this.dialogElement( "alwaysHandle" ).checked;
  601.  
  602.           // One last special case: If the input "always ask" flag was false, then we always
  603.           // update.  In that case we are displaying the helper app dialog for the first
  604.           // time for this mime type and we need to store the user's action in the mimeTypes.rdf
  605.           // data source (whether that action has changed or not; if it didn't change, then we need
  606.           // to store the "always ask" flag so the helper app dialog will or won't display
  607.           // next time, per the user's selection).
  608.           needUpdate = needUpdate || !this.mLauncher.MIMEInfo.alwaysAskBeforeHandling;
  609.  
  610.           // Make sure mime info has updated setting for the "always ask" flag.
  611.           this.mLauncher.MIMEInfo.alwaysAskBeforeHandling = !this.dialogElement( "alwaysHandle" ).checked;
  612.         }
  613.  
  614.         return needUpdate;
  615.     },
  616.  
  617.     // See if the user changed things, and if so, update the
  618.     // mimeTypes.rdf entry for this mime type.
  619.     updateHelperAppPref: function() {
  620.         // We update by passing this mime info into the "Edit Type" helper app
  621.         // pref dialog.  It will update the data source and close the dialog
  622.         // automatically.
  623.         this.mDialog.openDialog( "chrome://communicator/content/pref/pref-applications-edit.xul",
  624.                                  "_blank",
  625.                                  "chrome,modal=yes,resizable=no",
  626.                                  this );
  627.     },
  628.  
  629.     // onOK:
  630.     onOK: function() {
  631.         // Verify typed app path, if necessary.
  632.         if ( this.dialogElement( "openUsing" ).selected ) {
  633.             var helperApp = this.helperAppChoice();
  634.             if ( !helperApp || !helperApp.exists() ) {
  635.                 // Show alert and try again.
  636.                 var msg = this.replaceInsert( this.getString( "badApp" ), 1, this.dialogElement( "appPath" ).value );
  637.                 var svc = Components.classes[ "@mozilla.org/embedcomp/prompt-service;1" ]
  638.                             .getService( Components.interfaces.nsIPromptService );
  639.                 svc.alert( this.mDialog, this.getString( "badApp.title" ), msg );
  640.                 // Disable the OK button.
  641.                 this.mDialog.document.documentElement.getButton( "accept" ).disabled = true;
  642.                 // Select and focus the input field if input field is not disabled
  643.                 var path = this.dialogElement( "appPath" );
  644.                 if ( !path.disabled ) {
  645.                     path.select();
  646.                     path.focus();
  647.                 }
  648.                 // Clear chosen application.
  649.                 this.chosenApp = null;
  650.                 // Leave dialog up.
  651.                 return false;
  652.             }
  653.         }
  654.  
  655.         // Remove our web progress listener (a progress dialog will be
  656.         // taking over).
  657.         this.mLauncher.setWebProgressListener( null );
  658.  
  659.         // saveToDisk and launchWithApplication can return errors in
  660.         // certain circumstances (e.g. The user clicks cancel in the
  661.         // "Save to Disk" dialog. In those cases, we don't want to
  662.         // update the helper application preferences in the RDF file.
  663.         try {
  664.             var needUpdate = this.updateMIMEInfo();
  665.             if ( this.dialogElement( "saveToDisk" ).selected )
  666.                 this.mLauncher.saveToDisk( null, false );
  667.             else
  668.                 this.mLauncher.launchWithApplication( null, false );
  669.  
  670.             // Update user pref for this mime type (if necessary). We do not
  671.             // store anything in the mime type preferences for the ambiguous
  672.             // type application/octet-stream.
  673.             if ( needUpdate &&
  674.                  this.mLauncher.MIMEInfo.MIMEType != "application/octet-stream" )
  675.             {
  676.                 this.updateHelperAppPref();
  677.             }
  678.  
  679.         } catch(e) { }
  680.  
  681.         // Unhook dialog from this object.
  682.         this.mDialog.dialog = null;
  683.  
  684.         // Close up dialog by returning true.
  685.         return true;
  686.     },
  687.  
  688.     // onCancel:
  689.     onCancel: function() {
  690.         // Remove our web progress listener.
  691.         this.mLauncher.setWebProgressListener( null );
  692.  
  693.         // Cancel app launcher.
  694.         try {
  695.             const NS_BINDING_ABORTED = 0x804b0002;
  696.             this.mLauncher.cancel(NS_BINDING_ABORTED);
  697.         } catch( exception ) {
  698.         }
  699.  
  700.         // Unhook dialog from this object.
  701.         this.mDialog.dialog = null;
  702.  
  703.         // Close up dialog by returning true.
  704.         return true;
  705.     },
  706.  
  707.     // dialogElement:  Try cache; obtain from document if not there.
  708.     dialogElement: function( id ) {
  709.          // Check if we've already fetched it.
  710.          if ( !( id in this.elements ) ) {
  711.              // No, then get it from dialog.
  712.              this.elements[ id ] = this.mDialog.document.getElementById( id );
  713.          }
  714.          return this.elements[ id ];
  715.     },
  716.  
  717.     // chooseApp:  Open file picker and prompt user for application.
  718.     chooseApp: function() {
  719.         var nsIFilePicker = Components.interfaces.nsIFilePicker;
  720.         var fp = Components.classes["@mozilla.org/filepicker;1"].createInstance( nsIFilePicker );
  721.         fp.init( this.mDialog,
  722.                  this.getString( "chooseAppFilePickerTitle" ),
  723.                  nsIFilePicker.modeOpen );
  724.  
  725.         // XXX - We want to say nsIFilePicker.filterExecutable or something
  726.         fp.appendFilters( nsIFilePicker.filterAll );
  727.  
  728.         if ( fp.show() == nsIFilePicker.returnOK && fp.file ) {
  729.             // Remember the file they chose to run.
  730.             this.chosenApp = fp.file;
  731.             // Update dialog.
  732.             this.dialogElement( "appPath" ).value = this.getPath(this.chosenApp);
  733.         }
  734.     },
  735.  
  736.     // dumpInfo:
  737.     doDebug: function() {
  738.         const nsIProgressDialog = Components.interfaces.nsIProgressDialog;
  739.         // Open new progress dialog.
  740.         var progress = Components.classes[ "@mozilla.org/progressdialog;1" ]
  741.                          .createInstance( nsIProgressDialog );
  742.         // Show it.
  743.         progress.open( this.mDialog );
  744.     },
  745.  
  746.     // dumpObj:
  747.     dumpObj: function( spec ) {
  748.          var val = "<undefined>";
  749.          try {
  750.              val = eval( "this."+spec ).toString();
  751.          } catch( exception ) {
  752.          }
  753.          this.dump( spec + "=" + val + "\n" );
  754.     },
  755.  
  756.     // dumpObjectProperties
  757.     dumpObjectProperties: function( desc, obj ) {
  758.          for( prop in obj ) {
  759.              this.dump( desc + "." + prop + "=" );
  760.              var val = "<undefined>";
  761.              try {
  762.                  val = obj[ prop ];
  763.              } catch ( exception ) {
  764.              }
  765.              this.dump( val + "\n" );
  766.          }
  767.     },
  768.  
  769.     // getString: Fetch data string from dialog content (and cache it).
  770.     getString: function( id ) {
  771.         // Check if we've fetched this string already.
  772.         if ( !( id in this.strings ) ) {
  773.             // Try to get it.
  774.             var elem = this.mDialog.document.getElementById( id );
  775.             if ( elem
  776.                  &&
  777.                  elem.firstChild
  778.                  &&
  779.                  elem.firstChild.nodeValue ) {
  780.                 this.strings[ id ] = elem.firstChild.nodeValue;
  781.             } else {
  782.                 // If unable to fetch string, use an empty string.
  783.                 this.strings[ id ] = "";
  784.             }
  785.         }
  786.         return this.strings[ id ];
  787.     },
  788.  
  789.     // replaceInsert: Replace given insert with replacement text and return the result.
  790.     replaceInsert: function( text, insertNo, replacementText ) {
  791.         var result = text;
  792.         var regExp = new RegExp("#"+insertNo);
  793.         result = result.replace( regExp, replacementText );
  794.         return result;
  795.     }
  796. }
  797.  
  798. // This Component's module implementation.  All the code below is used to get this
  799. // component registered and accessible via XPCOM.
  800. var module = {
  801.     firstTime: true,
  802.  
  803.     // registerSelf: Register this component.
  804.     registerSelf: function (compMgr, fileSpec, location, type) {
  805.         if (this.firstTime) {
  806.             this.firstTime = false;
  807.             throw Components.results.NS_ERROR_FACTORY_REGISTER_AGAIN;
  808.         }
  809.         compMgr = compMgr.QueryInterface(Components.interfaces.nsIComponentRegistrar);
  810.  
  811.         compMgr.registerFactoryLocation( this.cid,
  812.                                          "Mozilla Helper App Launcher Dialog",
  813.                                          this.contractId,
  814.                                          fileSpec,
  815.                                          location,
  816.                                          type );
  817.     },
  818.  
  819.     // getClassObject: Return this component's factory object.
  820.     getClassObject: function (compMgr, cid, iid) {
  821.         if (!cid.equals(this.cid)) {
  822.             throw Components.results.NS_ERROR_NO_INTERFACE;
  823.         }
  824.  
  825.         if (!iid.equals(Components.interfaces.nsIFactory)) {
  826.             throw Components.results.NS_ERROR_NOT_IMPLEMENTED;
  827.         }
  828.  
  829.         return this.factory;
  830.     },
  831.  
  832.     /* CID for this class */
  833.     cid: Components.ID("{F68578EB-6EC2-4169-AE19-8C6243F0ABE1}"),
  834.  
  835.     /* Contract ID for this class */
  836.     contractId: "@mozilla.org/helperapplauncherdialog;1",
  837.  
  838.     /* factory object */
  839.     factory: {
  840.         // createInstance: Return a new nsProgressDialog object.
  841.         createInstance: function (outer, iid) {
  842.             if (outer != null)
  843.                 throw Components.results.NS_ERROR_NO_AGGREGATION;
  844.  
  845.             return (new nsHelperAppDialog()).QueryInterface(iid);
  846.         }
  847.     },
  848.  
  849.     // canUnload: n/a (returns true)
  850.     canUnload: function(compMgr) {
  851.         return true;
  852.     }
  853. };
  854.  
  855. // NSGetModule: Return the nsIModule object.
  856. function NSGetModule(compMgr, fileSpec) {
  857.     return module;
  858. }
  859.  
  860. // Since we're automatically downloading, we don't get the file picker's
  861. // logic to check for existing files, so we need to do that here.
  862. //
  863. // Note - this code is identical to that in contentAreaUtils.js.
  864. // If you are updating this code, update that code too! We can't share code
  865. // here since this is called in a js component.
  866. function uniqueFile(aLocalFile) {
  867.     while (aLocalFile.exists()) {
  868.         parts = /(-\d+)?(\.[^.]+)?$/.test(aLocalFile.leafName);
  869.         aLocalFile.leafName = RegExp.leftContext + (RegExp.$1 - 1) + RegExp.$2;
  870.     }
  871.     return aLocalFile;
  872. }
  873.